-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathzad3.py
72 lines (57 loc) · 1.78 KB
/
zad3.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
from zad3testy import runtests
def binary_search_first(arr: 'sorted sequence', el: 'searched element') -> int:
left_idx = 0
right_idx = len(arr) - 1
while left_idx <= right_idx:
mid_idx = (left_idx + right_idx) // 2
if el > arr[mid_idx]:
left_idx = mid_idx + 1
else:
right_idx = mid_idx - 1
return left_idx if left_idx < len(arr) and arr[left_idx] == el else -1
def insert_element(arr, val: 'inserted value'):
arr.append(val)
# Move all elements that are greater than a value inserted to the right
idx = len(arr) - 1
while idx > 0 and arr[idx - 1] > val:
arr[idx] = arr[idx - 1]
idx -= 1
# Place our value on the final position
arr[idx] = val
def longest_incomplete(A, k):
if k < 2: return 0
# Find all unique values
U = []
for val in A:
i = binary_search_first(U, val)
if i < 0:
insert_element(U, val)
# Using two pointers, look for the longest incomplete subsequence
i = j = 0
C = [0] * k
# Find first subsequence of k - 1 unique elements
remaining = k
while i < len(A) and remaining:
idx = binary_search_first(U, A[i])
if not C[idx]:
remaining -= 1
C[idx] += 1
i += 1
# Move the window of k - 1 unique elements
max_len = i - 1
while i < len(A):
if not remaining:
idx = binary_search_first(U, A[j])
C[idx] -= 1
if not C[idx]:
remaining += 1
j += 1
else:
idx = binary_search_first(U, A[i])
if not C[idx]:
remaining -= 1
max_len = max(max_len, i - j)
C[idx] += 1
i += 1
return max_len
runtests(longest_incomplete)